### Universidade Federal de Minas Gerais Instituto de Ciências Exatas Departamento de Ciência da Computação

DCC819 - Arquitetura de Computadores

Relat'orio~I - Unidade~L'ogica~Aritm'etica

Guilherme Batista Santos Iuri Silva Castro João Mateus de Freitas Veneroso Ricardo Pagoto Marinho

> Belo Horizonte - MG 10 de outubro de 2017

### 1 Introdução

O presente trabalho foca na descrição da implementação de uma Unidade Lógica e Aritmética (ULA) em Verilog HDL, uma Linguagem de descrição de hardware, Hardware Description Language - HDL - em inglês. A ULA é um circuito digital responsável por realizar operações lógicas e aritméticas no caminho de dados de uma CPU. É a ULA que realiza operações como adição, subtração e operações lógicas como and e or. Além disso, ela também é responsável por calcular o endereço de memória para escrita ou leitura quando as instruções requisitam. Para implementar a ALU, utilizamos a IDE Quartus II 13 junto com ModelSim para simular uma FPGA e fazer os testes necessários. Os testes em dispositivo físico foi feito no módulo de prototipação DE2.

#### 2 ULA

A Figura 1 mostra o desenho esquemático de uma ULA.



Figura 1: Desenho esquemático de uma ULA.

Nela, é possível ver que a ULA possui duas entradas e uma saída de dados, A, B e R respectivamente, além de uma entrada e uma saída de sinais de controle, F e D. As entradas A e B são os valores que a ULA recebe para fazer os cálculos necessários. Esses valores podem vir de registradores ou podem ser imediatos: números absolutos, como nas instruções a seguir:

```
Addi R1,R2,10
Sub R3,R4,R5
```

A instrução 1 soma o valor imediato 10 ao valor armazenado no registrador R2 e armazena no registrador R1. No caso dessa instrução, a entrada A na Figura 1 recebe o valor do registrador R2, a entrada B o valor do imediato (10) e a saída R terá o valor da soma, sendo que este valor será armazenado no registrador R1. Já a instrução 2 subtrai o valor armazenado no registrador R5 do valor armazenado em R4 e armazena em R3, fazendo a operação

$$R3 = R4 - R5 \tag{1}$$

A entrada de sinais F é a entrada de controles da ALU. Dentre esses sinais de controle está o que sinaliza se o valor na entrada de dados B é o valor vindo de um registrador ou um imediato. Esse

sinal controla um multiplexador para seleciona qual entrada irá passar para a ALU, a entrada vinda do banco de registradores ou a entrada vinda direto da instrução no caso de um imediato.

Já a saída de sinais D, são os sinais que a ALU indica, sendo eles se o resultado é zero, negativo e se houve overflow na operação.

## 3 Implementação

Esta seção fala da implementação da ULA em *Verilog*. Nela, mostramos como cada parte da ULA foi implementada além de decisões de projeto tomadas.

O módulo desenvolvido possui 5 entradas e duas saídas: *OpA*, *OpB*, *Op*, *RST*, *CLK*, *Res* e *FlagReg* respectivamente. O Apêndice 3 mostra o código.

As entradas OpA e OpB representam as entradas A e B da Figura 1. A entrada Op indica qual operação a ALU vai fazer (Add, Sub, etc.) e faz parte da entrada F na Figura 1. As saídas Res e FlagReg indicam, respectivamente o resultado da operação e o sinal de saída da ALU, na Figura 1, as saídas R e D respectivamente.

Na implementação, as entradas OpA e OpB são de 16 bits, que, de acordo com a especificação do trabalho, é o tamanho dos registradores. Aqui, apenas o OpB pode ser um imediato. Neste caso, o imediato possui apenas 4 bits, sendo necessário estender mais 12 bits. Mais a frente será mostrado como e onde essa operação é feita. A entrada Op indica a operação a ser realizada e possui 4 bits, ou seja, a ALU implementada possui um máximo de 16 operações diferentes. Pela especificação, apenas 11 operações foram utilizadas como mostrado na Tabela 1.

| Código | Operação | Descrição                      |
|--------|----------|--------------------------------|
| 0      | Add      | Adição com registradores       |
| 1      | Sub      | Subtração com registradores    |
| 2      | Slti     | Salto condicional com imediato |
| 3      | And      | And binário                    |
| 4      | Or       | Or binário                     |
| 5      | Xor      | Xor binário                    |
| 6      | Andi     | And binário com imediato       |
| 7      | Ori      | Or binário com imediato        |
| 8      | Xori     | Xor binário com imediato       |
| 9      | Addi     | Adição com imediato            |
| 10     | Subi     | Subtração com imediato         |

Tabela 1: Descrição das operações implementadas.

A saída Res possui 16 bits, assim como as entradas OpA e OpB já que é o resultado da operação e a saída FlagReg possui 3 bits, um para cada sinal de saída como mostra a Tabela 2.

| Índice | Sinal    |
|--------|----------|
| 0      | Overflow |
| 1      | Negativo |
| 2      | Zero     |

Tabela 2. Sinais de saída da ULA.

# **Apêndices**

Módulo ULA

```
module ULA (OpA, OpB, Res, Op, FlagReg, CLK, RST);
                             input CLK, RST;
                             input [3:0] Op;
 4
                             input [15:0] OpA, OpB;
 5
                             output reg [15:0] Res;
 6
                             output reg [2:0] FlagReg;
                                                                                                                                      // [Z N C]: Z=Zero; N=Neg;
                                     C=Carry/Overflow
                             wire [15:0] invOpB;
                             assign invOpB = ~OpB + 16'd1;
10
11
                             parameter InsADD = 4'b0000;
                                                                                                                 // ADD
                                                                                                                                      Res = OpA + OpB
12
                            parameter InsSUB = 4'b0001;
                                                                                                                 // SUB
                                                                                                                                      Res = OpA - OpB
13
                            parameter InsSLT = 4'b0010;
                                                                                                                 // SLTI Res = (OpA > OpB) ? 1 : 0
                             parameter InsAND = 4'b0011;
                                                                                                                 // AND
                                                                                                                                      Res = OpA & OpB
15
                             parameter InsOR = 4'b0100;
                                                                                                                 // OR
                                                                                                                                      Res = OpA \mid OpB
16
                             parameter InsXOR = 4'b0101;
                                                                                                                 // XOR
                                                                                                                                      Res = OpA ^ OpB
17
18
                             parameter OverflowFlag = 0;
19
                             parameter NegFlag
                                                                                                                 = 1;
20
                             parameter ZeroFlag
                                                                                                                 = 2;
21
22
                             always @(posedge CLK) begin
23
                               if (RST) begin
24
                                 FlagReg = 3'b000;
25
26
                               end
27
                               else begin
28
                                  case (Op)
                                     InsADD: begin
29
                                       Res = OpA + OpB;
30
31
                                       /* Zero check */
32
                                       if (Res == 0)
33
                                                    FlagReg[ZeroFlag] = 1'b1;
34
                                       else
35
                                         FlagReg[ZeroFlag] = 1'b0;
36
                                        /* Overflow check */
37
                                       FlagReg[OverflowFlag] = (OpA[15] & OpB[15] & ^Res[15]) | (^OpA[15] & ^Res[15] & ^Res[15]) | (^OpA[15] & ^Res[15] & ^Re
38
                                                15] & ~OpB[15] & Res[15]);
39
                                        /* Negative check */
                                       FlagReg[NegFlag] = Res[15];
40
                                     end
41
                                     InsSUB: begin
42
                                       Res = OpA + invOpB;
43
                                       /* Zero check */
44
                                       if (Res == 0)
45
                                         FlagReg[ZeroFlag] = 1'b1;
46
                                       else
47
                                          FlagReg[ZeroFlag] = 1'b0;
48
                                        /* Overflow check */
                                       FlagReg[OverflowFlag] = (OpA[15] & invOpB[15] & "Res[15]) | ("
50
                                                OpA[15] & ~invOpB[15] & Res[15]);
                                       /* Negative check */
51
```

```
FlagReg[NegFlag] = Res[15];
  52
   53
   54
                                                                                       InsSLT: begin
    55
                                                                                             if (OpA > OpB)
                                                                                                 Res = 16'd1;
   56
                                                                                             else
  57
                                                                                                 Res = 16'd0;
  58
                                                                                             /* Zero check */
  59
                                                                                            if (Res == 0)
   60
                                                                                                  FlagReg[ZeroFlag] = 1'b1;
   61
   62
                                                                                            else
                                                                                                  FlagReg[ZeroFlag] = 1'b0;
  63
                                                                                              /* Overflow check */
  64
                                                                                            FlagReg[OverflowFlag] = (OpA[15] & OpB[15] & ^Res[15]) | (^OpA[15] & ^Res[15] & ^Res[15]) | (^OpA[15] & ^Res[15] & ^Re
  65
                                                                                                                 15] & ~OpB[15] & Res[15]);
                                                                                              /* Negative check */
   66
                                                                                            FlagReg[NegFlag] = Res[15];
   67
                                                                                       end
  68
                                                                                       InsAND: begin
  69
                                                                                            Res = OpA & OpB;
  70
                                                                                            /* Zero check
   71
                                                                                            if (Res == 0)
   72
                                                                                                 FlagReg[ZeroFlag] = 1'b1;
  73
  74
                                                                                            else
                                                                                                 FlagReg[ZeroFlag] = 1'b0;
   75
                                                                                               /* Overflow check */
   76
                                                                                            FlagReg[OverflowFlag] = (OpA[15] & OpB[15] & ^Res[15]) | (^OpA[15] & ^Res[15] & ^Res[
   77
                                                                                                                 15] & ~OpB[15] & Res[15]);
                                                                                             /* Negative check */
   78
                                                                                            FlagReg[NegFlag] = Res[15];
   79
                                                                                       end
  80
                                                                                       InsOR: begin
  81
                                                                                            Res = OpA \mid OpB;
   82
                                                                                             /* Zero check */
                                                                                            if (Res == 0)
   84
                                                                                                 FlagReg[ZeroFlag] = 1'b1;
   85
                                                                                            else
  86
                                                                                                 FlagReg[ZeroFlag] = 1'b0;
  87
                                                                                              /* Overflow check */
   88
                                                                                            FlagReg[OverflowFlag] = (OpA[15] & OpB[15] & Res[15]) | (OpA[15] & FlagReg[OverflowFlag] | (OpA[15] & FlagReg[OverflowFlag] | (OpA[15] & FlagReg[OverflowFlag] | (OpA[15] & OpB[15] & OpB[15] & FlagReg[OverflowFlag] | (OpA[15] & OpB[15] & Op
                                                                                                                15] & ~OpB[15] & Res[15]);
                                                                                              /* Negative check */
  90
                                                                                            FlagReg[NegFlag] = Res[15];
  91
  92
                                                                                       InsXOR: begin
   93
                                                                                            Res = OpA 	 OpB;
   94
                                                                                             /* Zero check */
  95
                                                                                            if (Res == 0)
  96
                                                                                                 FlagReg[ZeroFlag] = 1'b1;
  97
  98
                                                                                                  FlagReg[ZeroFlag] = 1'b0;
   99
                                                                                              /* Overflow check */
 100
                                                                                            FlagReg[OverflowFlag] = (OpA[15] & OpB[15] & ~Res[15]) | (~OpA[
101
                                                                                                                15] & ~OpB[15] & Res[15]);
                                                                                             /* Negative check */
102
                                                                                            FlagReg[NegFlag] = Res[15];
103
104
                                                                                       end
                                                                                 endcase
```

```
106 end
107 end
108 endmodule
```

# Referências